查看原文
其他

Java并发:Semaphore信号量源码分析

狂小白 搜云库技术团队 2019-04-07
搜云库互联网/架构/开发/运维关注

JUC 中 Semaphore 的使用与原理分析,Semaphore 也是 Java 中的一个同步器,与 CountDownLatch 和 CycleBarrier 不同在于它内部的计数器是递增的,那么,Semaphore 的内部实现是怎样的呢?

Semaphore 信号量也是Java 中一个同步容器,与CountDownLatch 和 CyclicBarrier 不同之处在于它内部的计数器是递增的。为了能够一览Semaphore的内部结构,我们首先要看一下Semaphore的类图,类图,如下所示:

如上类图可以知道Semaphoren内部还是使用AQS来实现的,Sync只是对AQS的一个修饰,并且Sync有两个实现类,分别代表获取信号量的时候是否采取公平策略。创建Semaphore的时候会有一个变量标示是否使用公平策略,源码如下:

  1. public Semaphore(int permits) {

  2.    sync = new NonfairSync(permits);

  3. }

  4. public Semaphore(int permits, boolean fair) {

  5.    sync = fair ? new FairSync(permits) : new      

  6.    NonfairSync(permits);

  7. }

  8. Sync(int permits) {

  9.    setState(permits);

  10. }

如上面代码所示,Semaphore默认使用的是非公平策略,如果你需要公平策略,则可以使用带两个参数的构造函数来构造Semaphore对象,另外和CountDownLatch一样,构造函数里面传递的初始化信号量个数 permits 被赋值给了AQS 的state状态变量,也就是说这里AQS的state值表示当前持有的信号量个数。

接下来我们主要看看Semaphore实现的主要方法的源码,如下:

1、void acquire() 当前线程调用该方法的时候,目的是希望获取一个信号量资源,如果当前信号量计数个数大于 0 ,并且当前线程获取到了一个信号量则该方法直接返回,当前信号量的计数会减少 1 。否则会被放入AQS的阻塞队列,当前线程被挂起,直到其他线程调用了release方法释放了信号量,并且当前线程通过竞争获取到了改信号量。当前线程被其他线程调用了 interrupte()方法中断后,当前线程会抛出 InterruptedException异常返回。源码如下:

  1. public void acquire() throws InterruptedException {

  2.        //传递参数为1,说明要获取1个信号量资源

  3.        sync.acquireSharedInterruptibly(1);

  4.   }

  5. public final void acquireSharedInterruptibly(int arg)

  6.   throws InterruptedException {

  7. //(1)如果线程被中断,则抛出中断异常

  8. if (Thread.interrupted())

  9. throw new InterruptedException();

  10. //(2)否者调用sync子类方法尝试获取,这里根据构造函数确定使用公平策略

  11. if (tryAcquireShared(arg) < 0)

  12.    //如果获取失败则放入阻塞队列,然后再次尝试如果失败则调用park方法挂起当前线程

  13.    doAcquireSharedInterruptibly(arg);

  14. }

如上代码可知,acquire()内部调用了sync的acquireSharedInterruptibly  方法,后者是对中断响应的(如果当前线程被中断,则抛出中断异常),尝试获取信号量资源的AQS的方法tryAcquireShared 是由 sync 的子类实现,所以这里就要分公平性了,这里先讨论非公平策略 NonfairSync 类的 tryAcquireShared 方法,源码如下:

  1. protected int tryAcquireShared(int acquires) {

  2.    return nonfairTryAcquireShared(acquires);

  3. }

  4. final int nonfairTryAcquireShared(int acquires) {

  5.    for (;;) {

  6.     //获取当前信号量值

  7.     int available = getState();

  8.     //计算当前剩余值

  9.     int remaining = available - acquires;

  10.     //如果当前剩余小于0或者CAS设置成功则返回

  11.     if (remaining < 0 ||

  12.         compareAndSetState(available, remaining))

  13.         return remaining;

  14.    }

  15. }

如上代码,先计算当前信号量值(available)减去需要获取的值(acquires) 得到剩余的信号量个数(remaining),如果剩余值小于 0 说明当前信号量个数满足不了需求,则直接返回负数,然后当前线程会被放入AQS的阻塞队列,当前线程被挂起。如果剩余值大于 0 则使用CAS操作设置当前信号量值为剩余值,然后返回剩余值。另外可以知道NonFairSync是非公平性获取的,是说先调用aquire方法获取信号量的线程不一定比后来者先获取锁。

接下来我们要看看公平性的FairSync 类是如何保证公平性的,源码如下:

  1. protected int tryAcquireShared(int acquires) {

  2.        for (;;) {

  3.            if (hasQueuedPredecessors())

  4.                return -1;

  5.            int available = getState();

  6.            int remaining = available - acquires;

  7.            if (remaining < 0 || compareAndSetState(available, remaining))

  8.                return remaining;

  9.        }

  10. }

可以知道公平性还是靠 hasQueuedPredecessors 这个方法来做的,以前的随笔已经讲过公平性是看当前线程节点是否有前驱节点也在等待获取该资源,如果是则自己放弃获取的权力,然后当前线程会被放入AQS阻塞队列,否则就去获取。hasQueuedPredecessors源码如下:

  1. public final boolean hasQueuedPredecessors() {

  2.        Node t = tail;

  3.        Node h = head;

  4.        Node s;

  5.        return h != t && ((s = h.next) == null || s.thread != Thread.currentThread());

  6. }

如上面代码所示,如果当前线程节点有前驱节点则返回true,否则如果当前AQS队列为空 或者 当前线程节点是AQS的第一个节点则返回 false ,其中,如果 h == t 则说明当前队列为空则直接返回 false,如果 h !=t 并且 s == null 说明有一个元素将要作为AQS的第一个节点入队列(回顾下 enq 函数第一个元素入队列是两步操作,首先创建一个哨兵头节点,然后第一个元素插入到哨兵节点后面),那么返回 true,如果  h !=t 并且 s != null 并且  s.thread != Thread.currentThread() 则说明队列里面的第一个元素不是当前线程则返回 true。

2、void acquire(int permits) 该方法与 acquire() 不同在与后者只需要获取一个信号量值,而前者则获取指定 permits 个,源码如下:

  1. public void acquire(int permits) throws InterruptedException {

  2.        if (permits < 0)

  3.        throw new IllegalArgumentException();

  4.        sync.acquireSharedInterruptibly(permits);

  5. }

3、void acquireUninterruptibly() 该方法与 acquire() 类似,不同之处在于该方法对中断不响应,也就是当当前线程调用了 acquireUninterruptibly 获取资源过程中(包含被阻塞后)其它线程调用了当前线程的 interrupt()方法设置了当前线程的中断标志当前线程并不会抛出 InterruptedException 异常而返回。源码如下:

  1. public void acquireUninterruptibly() {

  2.     sync.acquireShared(1);

  3. }

4、void acquireUninterruptibly(int permits) 该方法与 acquire(int permits) 不同在于该方法对中断不响应。源码如如下:

  1. public void acquireUninterruptibly(int permits) {

  2.        if (permits < 0) throw new IllegalArgumentException();

  3.        sync.acquireShared(permits);

  4. }

5、void release() 该方法作用是把当前 semaphore对象的信号量值增加 1 ,如果当前有线程因为调用 acquire 方法被阻塞放入了 AQS的阻塞队列,则会根据公平策略选择一个线程进行激活,激活的线程会尝试获取刚增加的信号量,源码如下:

  1. public void release() {

  2.        //(1)arg=1

  3.        sync.releaseShared(1);

  4.    }

  5.    public final boolean releaseShared(int arg) {

  6.        //(2)尝试释放资源

  7.        if (tryReleaseShared(arg)) {

  8.            //(3)资源释放成功则调用park唤醒AQS队列里面最先挂起的线程

  9.            doReleaseShared();

  10.            return true;

  11.        }

  12.        return false;

  13.    }

  14.    protected final boolean tryReleaseShared(int releases) {

  15.        for (;;) {

  16.            //(4)获取当前信号量值

  17.            int current = getState();

  18.            //(5)当前信号量值增加releases,这里为增加1

  19.            int next = current + releases;

  20.            if (next < current) // 移除处理

  21.                throw new Error("Maximum permit count exceeded");

  22.            //(6)使用cas保证更新信号量值的原子性

  23.            if (compareAndSetState(current, next))

  24.                return true;

  25.        }

  26.    }

如上面代码可以看到 release()方法中对 sync.releaseShared(1),可以知道release方法每次只会对信号量值增加 1 ,tryReleaseShared方法是无限循环,使用CAS保证了 release 方法对信号量递增 1 的原子性操作。当tryReleaseShared 方法增加信号量成功后会执行代码(3),调用AQS的方法来激活因为调用acquire方法而被阻塞的线程。

6、void release(int permits) 该方法与不带参数的不同之处在于前者每次调用会在信号量值原来基础上增加 permits,而后者每次增加 1。源码如下:

  1. public void release(int permits) {

  2.        if (permits < 0) throw new IllegalArgumentException();

  3.        sync.releaseShared(permits);

  4. }

另外注意到这里调用的是 sync.releaseShared 是共享方法,这说明该信号量是线程共享的,信号量没有和固定线程绑定,多个线程可以同时使用CAS去更新信号量的值而不会阻塞。

到目前已经知道了其原理,接下来用一个例子来加深对Semaphore的理解,例子如下:

  1. package com.hjc;

  2. import java.util.concurrent.ExecutorService;

  3. import java.util.concurrent.Executors;

  4. import java.util.concurrent.Semaphore;

  5. /**

  6. * Created by cong on 2018/7/8.

  7. */

  8. public class SemaphoreTest {

  9.    // 创建一个Semaphore实例

  10.    private static volatile Semaphore semaphore = new Semaphore(0);

  11.    public static void main(String[] args) throws InterruptedException {

  12.        ExecutorService executorService = Executors.newFixedThreadPool(2);

  13.        // 加入线程A到线程池

  14.        executorService.submit(new Runnable() {

  15.            public void run() {

  16.                try {

  17.                    System.out.println(Thread.currentThread() +  " over");

  18.                    semaphore.release();

  19.                } catch (Exception e) {

  20.                    e.printStackTrace();

  21.                }

  22.            }

  23.        });

  24.        // 加入线程B到线程池

  25.        executorService.submit(new Runnable() {

  26.            public void run() {

  27.                try {

  28.                    System.out.println(Thread.currentThread() +  " over");

  29.                    semaphore.release();

  30.                } catch (Exception e) {

  31.                    e.printStackTrace();

  32.                }

  33.            }

  34.        });

  35.        // 等待子线程执行完毕,返回

  36.        semaphore.acquire(2);

  37.        System.out.println("all child thread over!");

  38.        //关闭线程池

  39.        executorService.shutdown();

  40.    }

  41. }

运行结果如下:

  1. Thread[pool-1-thread-1,5,main] over

  2. Thread[pool-1-thread-2,5,main] over

  3. all child thread over!

类似于 CountDownLatch,上面我们的例子也是在主线程中开启两个子线程进行执行,等所有子线程执行完毕后主线程在继续向下运行。

如上代码首先首先创建了一个信号量实例,构造函数的入参为 0,说明当前信号量计数器为 0,然后 main 函数添加两个线程任务到线程池,每个线程内部调用了信号量的 release 方法,相当于计数值递增一,最后在 main 线程里面调用信号量的 acquire 方法,参数传递为 2 说明调用 acquire 方法的线程会一直阻塞,直到信号量的计数变为 2 时才会返回。

看到这里也就明白了,如果构造 Semaphore 时候传递的参数为 N,在 M 个线程中调用了该信号量的 release 方法,那么在调用 acquire 对 M 个线程进行同步时候传递的参数应该是 M+N;

对CountDownLatch,CyclicBarrier,Semaphored这三者之间的比较总结:

1、CountDownLatch 通过计数器提供了更灵活的控制,只要检测到计数器为 0,而不管当前线程是否结束调用 await 的线程就可以往下执行,相比使用 jion 必须等待线程执行完毕后主线程才会继续向下运行更灵活。

2、CyclicBarrier 也可以达到 CountDownLatch 的效果,但是后者当计数器变为 0 后,就不能在被复用,而前者则使用 reset 方法可以重置后复用,前者对同一个算法但是输入参数不同的类似场景下比较适用。

3、而 semaphore 采用了信号量递增的策略,一开始并不需要关心需要同步的线程个数,等调用 aquire 时候在指定需要同步个数,并且提供了获取信号量的公平性策略。

▼往期精彩回顾▼

Mybatis 一级缓存清理无效引起的源码走读

Dubbo 整合 Pinpoint 做分布式服务请求跟踪

接口限流:漏桶算法&令牌桶算法

Java并发:深入浅出AQS之共享锁模式源码分析

Java并发:深入浅出AQS之独占锁模式源码分析

Java并发:了解无锁CAS就从源码分析

Java并发:CAS原理分析

长按:二维码关注

专注于开发技术研究与知识分享

公众号回复:”进群” 加微信群深入交流

回复 JavaDocker等关键字可获得学习资料

    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存